2 // @name Thread Rebuilder
3 // @namespace http://tampermonkey.net/
5 // @description try to take over the world!
6 // @author ECHibiki /qa/
7 // @match https://boards.4chan.org/*/thread/*
8 // @match http://boards.4chan.org/*/thread/*
9 // @grant GM_xmlhttpRequest
10 // @updateURL https://github.com/ECHibiki/4chan-UserScripts/raw/master/Thread-Rebuilder.user.js
11 // @downloadURL https://github.com/ECHibiki/4chan-UserScripts/raw/master/Thread-Rebuilder.user.js
12 // @run-at document-start
16 var thread_data = [['Comment'], ['Image URLs'], ['Image Names'] ,['Post No.']];
18 var semaphore_posts = 1;
21 var use_offsite_archive = false;
22 var window_displayed = false;
23 var in_sequence = false;
24 var tool_top_visible = false;
26 var help_icon_source = " ";
29 //set listener to build interface in 4chanX
30 //set listeners to build interface in 4chanX
31 document.addEventListener("4chanXInitFinished", function(e){
32 document.addEventListener("QRDialogCreation", enhance4ChanX);
37 use_offsite_archive = localStorage.getItem("ArchiveType") == 0 ? true : false;
38 if(use_offsite_archive) document.getElementById("OffsiteArchive").checked = true;
39 else document.getElementById("OnsiteArchive").checked = true;
45 function storageAvailable(type) {
47 var storage = window[type],
48 x = '__storage_test__';
49 storage.setItem(x, x);
50 storage.removeItem(x);
54 //From https://developer.mozilla.org/en-US/docs/Web/API/Web_Storage_API/Using_the_Web_Storage_API
55 return e instanceof DOMException && (
56 // everything except Firefox
60 // test name field too, because code might not be present
61 // everything except Firefox
62 e.name === 'QuotaExceededError' ||
64 e.name === 'NS_ERROR_DOM_QUOTA_REACHED') &&
65 // acknowledge QuotaExceededError only if there's something already stored
70 //settings for time expiration on image hiding
71 function rebuildWindow(){
72 var style = document.createElement('style');
73 style.innerHTML = ".inputs{background-color:rgb(200,200,200);margin:5px 7px;width:100px;}";
74 document.body.appendChild(style);
76 var background_div = document.createElement("div");
77 background_div.setAttribute("style", "border:solid 1px black;position:fixed;width:100%;height:100%;background-color:rgba(200,200,200,0.3);top:0;left:0;display:none; z-index:9");
78 background_div.setAttribute("id", "rebuildBackground");
79 document.body.appendChild(background_div);
80 background_div.addEventListener("click", rebuildToggle);
82 var window_div = document.createElement("div");
83 window_div.setAttribute("style", "border:solid 1px black;position:fixed;width:400px;background-color:rgb(200,200,200);left:40%;top:20%;margin-bottom:0; display:none; z-index:10");
84 window_div.setAttribute("id", "rebuildWindow");
86 var close_div = document.createElement("div");
87 close_div.setAttribute("style", "border:solid 1px black;position:absolute;width:25px;height:25px;background-color:rgba(255,100,90,0.9); right:3px;top:3px; z-index:10");
88 close_div.addEventListener("click", rebuildToggle);
89 window_div.appendChild(close_div);
91 var title_para = document.createElement("p");
92 title_para.setAttribute("style", "margin-left:5px;margin-top:5px");
93 var title_text = document.createTextNode("Rebuild Settings");
94 title_para.appendChild(title_text);
95 window_div.appendChild(title_para);
97 var container_div = document.createElement("div");
98 container_div.setAttribute("style","background-color:white;margin:0 0;padding:5px;");
99 window_div.appendChild(container_div);
101 var rebuild_label_local = document.createElement("label");
102 var rebuild_text_local = document.createTextNode("Use 4chan Archives: ");
103 rebuild_label_local.appendChild(rebuild_text_local);
104 container_div.appendChild(rebuild_label_local);
105 var rebuild_input_local = document.createElement("input");
106 rebuild_input_local.setAttribute("type", "radio");
107 rebuild_input_local.setAttribute("name", "ArchiveSettings");
108 rebuild_input_local.setAttribute("id", "OnsiteArchive");
109 container_div.appendChild(rebuild_input_local);
110 container_div.appendChild(rebuild_input_local);
111 container_div.appendChild(document.createElement("br"));
113 var rebuild_label_offsite = document.createElement("label");
114 var rebuild_text_offsite = document.createTextNode("Use Offsite Archives: ");
115 rebuild_label_offsite.appendChild(rebuild_text_offsite);
116 container_div.appendChild(rebuild_label_offsite);
117 var rebuild_input_offsite = document.createElement("input");
118 rebuild_input_offsite.setAttribute("type", "radio");
119 rebuild_input_offsite.setAttribute("name", "ArchiveSettings");
120 rebuild_input_offsite.setAttribute("id", "OffsiteArchive");
121 container_div.appendChild(rebuild_input_offsite);
122 container_div.appendChild(rebuild_input_offsite);
123 container_div.appendChild(document.createElement("br"));
125 var set_button = document.createElement("input");
126 set_button.setAttribute("type", "button");
127 set_button.setAttribute("id", "setTime");
128 set_button.setAttribute("value", "Set Archive");
129 set_button.addEventListener("click", function(){
130 if (storageAvailable('localStorage')) {
131 var radio_options = document.getElementsByName("ArchiveSettings");
132 for (var radio_input = 0 ; radio_input < radio_options.length; radio_input++)
133 if(radio_options[radio_input].checked){
134 localStorage.setItem("ArchiveType", radio_input);
135 if(radio_input == 0) use_offsite_archive = true;
140 container_div.appendChild(set_button);
142 document.body.appendChild(window_div);
146 function rebuildToggle(){
147 if(window_displayed){
148 document.getElementById("rebuildWindow").style.display = "none";
149 document.getElementById("rebuildBackground").style.display = "none";
150 window_displayed = false;
153 document.getElementById("rebuildWindow").style.display = "inline-block";
154 document.getElementById("rebuildBackground").style.display = "inline-block";
155 window_displayed = true;
159 function rebuildButton(){
160 var rebuild_button = document.createElement("input");
161 rebuild_button.setAttribute("Value", "Thread Rebuilder Settings");
162 rebuild_button.setAttribute("type", "button");
163 rebuild_button.setAttribute("style", "position:absolute;top:105px");
164 rebuild_button.addEventListener("click", rebuildWindow);
165 if(document.body === null){
166 setTimeout(rebuildButton, 30);
169 document.body.appendChild(rebuild_button);
170 rebuild_button.addEventListener("click", rebuildToggle);
174 var enhance4ChanX = function(){
175 var qr_window = document.getElementById("qr");
177 if(document.getElementById("qrRebuilder") !== null) qr_window.removeChild(document.getElementById("qrRebuilder"));
179 var thread_rebuilder_table = document.createElement("TABLE");
180 thread_rebuilder_table.setAttribute("id", "qrRebuilder");
181 thread_rebuilder_table.setAttribute("style", "text-align:center");
182 qr_window.appendChild(thread_rebuilder_table);
184 var thread_row = document.createElement("TR");
186 var help_icon_container = document.createElement("A");
187 help_icon_container.href = "javascript:void(0)";
188 help_icon_container.title = "Click to View Help!";
189 var help_icon = document.createElement("IMG");
190 help_icon.setAttribute("style", "height:" + option_text_size * 1.25 + "px;margin:-4px 10px");
191 help_icon.src = help_icon_source;
193 help_icon_container.appendChild(help_icon);
194 thread_row.appendChild(help_icon_container);
196 var tooltip_div = document.createElement("DIV");
197 tooltip_div.innerHTML = "Insert the thread number of the post to rebuild<br/>Must be in either the 4chan archives or archived.moe<hr/>Submit bugs to <a href='https://github.com/ECHibiki/4chan-UserScripts'>my Github</a>";
198 tooltip_div.setAttribute("style", "z-index:9;padding:5px;border:1px solid black;background-color:white;word-wrap:break-word;display:none;position:absolute;");
199 help_icon_container.addEventListener("click", function(ev){
201 tooltip_div.setAttribute("style", "z-index:9;padding:5px;border:1px solid black;background-color:white;word-wrap:break-word;display:none;position:absolute;");
203 tooltip_div.setAttribute("style", "z-index:9;padding:5px;border:1px solid black;background-color:white;word-wrap:break-word;display:block;position:absolute;"
204 + "left:" + (ev.clientX - qr_window.getBoundingClientRect().x) +
205 "px;top:" + (ev.clientY - qr_window.getBoundingClientRect().y ) + "px;");
206 tool_top_visible = !tool_top_visible;
208 qr_window.appendChild(tooltip_div);
210 var second_row_nodes = [
211 document.createTextNode("Thread: "),
212 document.createElement("INPUT"),
213 document.createElement("INPUT"),
215 second_row_nodes.forEach(
217 thread_row.appendChild(node);
219 thread_rebuilder_table.appendChild(thread_row);
221 second_row_nodes[1].setAttribute("ID", "threadInput");
222 second_row_nodes[1].setAttribute("style", "width:35.0%");
224 second_row_nodes[2].setAttribute("ID", "threadButton");
225 second_row_nodes[2].setAttribute("type", "button");
226 second_row_nodes[2].setAttribute("value", "Set Rebuild Queue");
228 second_row_nodes[2].addEventListener("click", function(){
231 getThread(second_row_nodes[1].value);
232 postID = setInterval(postRoutine, 1000);
233 if(timeListen === undefined) timeListen = setInterval(timeListenerFunction, 1000);
235 qr_window.appendChild(document.createElement("hr"));
238 var thread_data_length = 0;
239 var posts_created = 0;
241 var postRoutine = function(){
244 thread_data_length = thread_data[0].length;
245 fillID = setInterval(fillRoutine, 10);
250 var stopRoutine = function(){
251 clearInterval(postID);
255 var fillRoutine = function(){
256 if(posts_created >= thread_data_length) {semaphore_posts = 0 ; stopFillRoutine();}
257 else if(semaphore_posts == 1){
259 createPost(thread_data[0][posts_created], thread_data[1][posts_created], thread_data[2][posts_created]);
264 var stopFillRoutine = function(){
265 clearInterval(fillID);
268 var setPropperLinking = function(text){
269 var search_regex = RegExp(">>\\d+", "g");
272 var link_arr = Array();
273 while((result = search_regex.exec(text)) != null){
274 var end_index = search_regex.lastIndex;
275 var post_no = result.toString().replace(/>/g, "");
276 link_arr.push([post_no, end_index]);
278 //hunt down the text of what it linked to
279 //Get the links inside of the origonal message to show text contents
280 var responding_text = Array();
281 if(use_offsite_archive)
282 URL = "https://www.archived.moe/_/api/chan/thread/?board=" + board + "&num=" + document.getElementById("threadInput").value;
284 URL = "https://a.4cdn.org/" + board + "/thread/" + document.getElementById("threadInput").value + ".json";
285 var xhr = new GM_xmlhttpRequest(({
288 responseType : "json",
289 onload: function(data){
290 if(use_offsite_archive)
291 data = data.response["" + document.getElementById("threadInput").value]["posts"];
293 data = data.response["posts"];
294 if(data == undefined){
295 alert("Invalid Thread ID: " + document.getElementById("threadInput").value + ". ");
298 link_arr.forEach(function(link_item){
299 for(var data_entry = 0 ; data_entry < data.length ; data_entry++){
300 if(parseInt(link_item[0]) == parseInt(data[data_entry]["no"])){
301 if(use_offsite_archive)
302 responding_text.push([ [post_no, end_index], data[data_entry]["comment_processed"].replace(/(>>|https:\/\/www\.archived\.moe\/.*\/thread\/.*\/#)\d+/g, ""), link_item["media"]["safe_media_hash"] ]);
304 responding_text.push([ [post_no, end_index], data[data_entry]["com"].replace(/(>>|#p)\d+/g, ""), data[data_entry]["md5"] ]);
310 var current_url = window.location.href;
311 var hash_index = current_url.lastIndexOf("#") != -1 ? current_url.lastIndexOf("#"): window.location.href.length;
312 var current_thread = window.location.href.substring(current_url.lastIndexOf("/")+1, hash_index);
313 var current_url = "https://a.4cdn.org/" + board + "/thread/" + current_thread + ".json";
314 //open current thread to hunt down the text found in links
315 var xhr = new GM_xmlhttpRequest(({
318 responseType : "json",
319 onload: function(data){
320 data = data.response["posts"];
321 if(data == undefined){
322 alert("Invalid Thread ID: " + document.getElementById("threadInput").value + ". ");
325 responding_text.forEach(function(response_item){
326 for(var data_entry = 0 ; data_entry < data.length ; data_entry++){
327 if((response_item[1] == data[data_entry]["com"].replace(/(>>|#p)\d+/g, "") || response_item[1] == null)
328 && (response_item[2] == data[data_entry]["md5"] || response_item[2] == null)){
329 var start_index = response_item[0][0].legth - response_item[0][1];
330 text = text.substring(0, start_index) + ">>" + data[data_entry]["no"] + text.substring(response_item[0][1]);
335 document.getElementById("qr").getElementsByTagName("TEXTAREA")[0].value = text;
336 document.getElementById("add-post").click();
347 //2) GET ARCHIVED THREAD
348 var getThread = function(threadNo){
349 thread_data = [[], [], [], []];
351 if(use_offsite_archive)
352 URL = "https://www.archived.moe/_/api/chan/thread/?board=" + board + "&num=" + document.getElementById("threadInput").value;
354 URL = "https://a.4cdn.org/" + board + "/thread/" + document.getElementById("threadInput").value + ".json";
355 var xhr = new GM_xmlhttpRequest(({
358 responseType : "json",
359 onload: function(data){
360 var starting_post = -1;
361 if(use_offsite_archive){
363 data = data.response["" + document.getElementById("threadInput").value];
367 data = data.response;
369 if(data == undefined){
370 alert("Invalid Thread ID: " + threadNo + ".\n4chan Archive ");
373 if(use_offsite_archive) data["posts"] = Object.values(data["posts"]);
374 var len = data["posts"].length;
376 for(var post_number = starting_post ; post_number < len ; post_number++){
377 var comment = undefined;
378 if(use_offsite_archive)
379 comment = data["posts"][post_number]["comment"];
381 comment = data["posts"][post_number]["com"];
382 if(comment !== undefined && comment !== null)
383 thread_data[0].push(comment);
385 thread_data[0].push(-1);
387 var filename = undefined;
388 if(use_offsite_archive)
389 if(data["posts"][post_number]["media"] !== null)
390 filename = "" + data["posts"][post_number]["media"]["media_filename"];
392 filename = "" + data["posts"][post_number]["tim"] + data["posts"][post_number]["ext"];
394 if(filename !== undefined && filename !== null && filename.indexOf("undefined") == -1)
395 if(use_offsite_archive)
396 if(data["posts"][post_number]["media"] !== null)
397 thread_data[1].push(data["posts"][post_number]["media"]["remote_media_link"]);
398 else thread_data[1].push(-1);
400 thread_data[1].push("https://i.4cdn.org/" + board + "/" + filename);
401 else thread_data[1].push(-1);
403 if(use_offsite_archive)
404 if(data["posts"][post_number]["media"] !== null)
405 thread_data[2].push(data["posts"][post_number]["media"]["media_id"]);
407 thread_data[2].push(data["posts"][post_number]["filename"]);
409 if(use_offsite_archive)
410 thread_data[3].push(data["posts"][post_number]["num"]);
412 thread_data[3].push(data["posts"][post_number]["no"]);
419 //3) RIP POSTS AND IMAGES
420 var createPost = function(text, imageURL, imageName){
422 var response_type = "arraybuffer";
423 if(use_offsite_archive) response_type = "text"
424 var xhr = new GM_xmlhttpRequest(({
427 responseType : response_type,
428 onload: function(response)
430 if(use_offsite_archive){
431 var parser = new DOMParser();
432 var content_attribute = parser.parseFromString(response.response, "text/html").getElementsByTagName("META")[0].getAttribute("content");
433 var redirect_url = content_attribute.substring(content_attribute.indexOf("http"));
434 var xhr = new GM_xmlhttpRequest(({method:"GET", url: redirect_url, responseType:"arraybuffer",
435 onload:function(response){
436 inputImage(response, text, imageURL, imageName);
441 inputImage(response, text, imageURL, imageName);
447 text = createPostComment(text);
448 setPropperLinking(text);
452 function inputImage(response, text, imageURL, imageName){
455 if(imageURL.indexOf(".jpg") > -1){
456 blob = new Blob([response.response], {type:"image/jpeg"});
459 else if(imageURL.indexOf(".png") > -1){
460 blob = new Blob([response.response], {type:"image/png"});
463 else if(imageURL.indexOf(".gif") > -1){
464 blob = new Blob([response.response], {type:"image/gif"});
467 else if(imageURL.indexOf(".webm") > -1){
468 blob = new Blob([response.response], {type:"video/webm"});
472 var name = imageName + ext;
474 //SEND RESULTING RESPONSE TO 4CHANX FILES === QRSetFile
475 var detail = {file:blob, name:name};
476 if (typeof cloneInto === 'function') {
477 detail = cloneInto(detail , document.defaultView);
480 document.dispatchEvent(new CustomEvent('QRSetFile', {bubbles:true, detail}));
482 if(text !== "" && text !== undefined && text !== -1) {
483 text = createPostComment(text);
484 setPropperLinking(text);
487 document.getElementById("add-post").click();
492 //4) CREATE POST QUEUE
493 var createPostComment = function(text){
494 text = text.replace(/<a href="\/[a-zA-Z]+\/" class="quotelink">/g, "");
495 text = text.replace(/<span class="deadlink">/g, "");
497 var quote_regex = /<a href="#p[0-9]+" class="quotelink">>>[0-9]+/g;
498 var find = text.match(quote_regex);
500 find.forEach(function(match){
501 var index_start = text.indexOf(match);
502 var match_len = match.length;
503 var index_len = index_start + match_len;
504 var first_quote = match.indexOf('"');
505 var second_quote = match.indexOf('"', first_quote + 1);
506 var post_no = match.substring(first_quote + 3, second_quote);
508 match = ">>" + post_no;
510 text = text.substr(0, index_start) + match + text.substr(index_len);
514 text = text.replace(/<span class="quote">/g, "");
515 text = text.replace(/<br>/g, "\n");
516 text = text.replace(/'/g, "'");
517 text = text.replace(/>/g, ">");
518 text = text.replace(/<\/a>/g, "");
519 text = text.replace(/<wbr>/g, "");
520 text = text.replace(/<\/span>/g, "");
526 var timeListenerFunction = function(){
527 var time = document.getElementById("qr-filename-container").nextSibling.value.replace(/[a-zA-Z]+/g, "");
536 document.addEventListener('QRPostSuccessful', function(e) {
538 document.getElementById("dump-list").childNodes[1].click();
539 setPropperLinking(document.getElementById("qr").getElementsByTagName("TEXTAREA")[0].value);
545 thread_data_length = 0;
553 thread_data = [['Comment'], ['Image URLs'], ['Image Names'] ,['Post No.']];
555 var qr_dumplist = document.getElementById("dump-list").childNodes;
556 var qr_dumplist_len = qr_dumplist.length;
557 var current_preview = 0;
558 while(qr_dumplist_len - current_preview > 1){
559 qr_dumplist[0].firstChild.click();